auto-coder 0.1.400__py3-none-any.whl → 2.0.0__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of auto-coder might be problematic. Click here for more details.
- auto_coder-2.0.0.dist-info/LICENSE +158 -0
- auto_coder-2.0.0.dist-info/METADATA +558 -0
- auto_coder-2.0.0.dist-info/RECORD +795 -0
- {auto_coder-0.1.400.dist-info → auto_coder-2.0.0.dist-info}/WHEEL +1 -1
- {auto_coder-0.1.400.dist-info → auto_coder-2.0.0.dist-info}/entry_points.txt +3 -3
- autocoder/__init__.py +31 -0
- autocoder/agent/auto_filegroup.py +32 -13
- autocoder/agent/auto_learn_from_commit.py +9 -1
- autocoder/agent/base_agentic/__init__.py +3 -0
- autocoder/agent/base_agentic/agent_hub.py +1 -1
- autocoder/agent/base_agentic/base_agent.py +235 -136
- autocoder/agent/base_agentic/default_tools.py +119 -118
- autocoder/agent/base_agentic/test_base_agent.py +1 -1
- autocoder/agent/base_agentic/tool_registry.py +32 -20
- autocoder/agent/base_agentic/tools/read_file_tool_resolver.py +25 -4
- autocoder/agent/base_agentic/tools/write_to_file_tool_resolver.py +24 -11
- autocoder/agent/base_agentic/types.py +42 -0
- autocoder/agent/entry_command_agent/chat.py +73 -59
- autocoder/auto_coder.py +31 -40
- autocoder/auto_coder_rag.py +11 -1084
- autocoder/auto_coder_runner.py +1029 -2310
- autocoder/auto_coder_terminal.py +26 -0
- autocoder/auto_coder_terminal_v3.py +190 -0
- autocoder/chat/conf_command.py +224 -124
- autocoder/chat/models_command.py +361 -299
- autocoder/chat/rules_command.py +79 -31
- autocoder/chat_auto_coder.py +1021 -372
- autocoder/chat_auto_coder_lang.py +23 -732
- autocoder/commands/auto_command.py +26 -9
- autocoder/commands/auto_web.py +1 -1
- autocoder/commands/tools.py +44 -44
- autocoder/common/__init__.py +150 -128
- autocoder/common/ac_style_command_parser/__init__.py +39 -2
- autocoder/common/ac_style_command_parser/config.py +422 -0
- autocoder/common/ac_style_command_parser/parser.py +292 -78
- autocoder/common/ac_style_command_parser/test_parser.py +241 -16
- autocoder/common/ac_style_command_parser/test_typed_parser.py +342 -0
- autocoder/common/ac_style_command_parser/typed_parser.py +653 -0
- autocoder/common/action_yml_file_manager.py +25 -13
- autocoder/common/agent_events/__init__.py +52 -0
- autocoder/common/agent_events/agent_event_emitter.py +193 -0
- autocoder/common/agent_events/event_factory.py +177 -0
- autocoder/common/agent_events/examples.py +307 -0
- autocoder/common/agent_events/types.py +113 -0
- autocoder/common/agent_events/utils.py +68 -0
- autocoder/common/agent_hooks/__init__.py +44 -0
- autocoder/common/agent_hooks/examples.py +582 -0
- autocoder/common/agent_hooks/hook_executor.py +217 -0
- autocoder/common/agent_hooks/hook_manager.py +288 -0
- autocoder/common/agent_hooks/types.py +133 -0
- autocoder/common/agent_hooks/utils.py +99 -0
- autocoder/common/agent_query_queue/queue_executor.py +324 -0
- autocoder/common/agent_query_queue/queue_manager.py +325 -0
- autocoder/common/agents/__init__.py +11 -0
- autocoder/common/agents/agent_manager.py +323 -0
- autocoder/common/agents/agent_parser.py +189 -0
- autocoder/common/agents/example_usage.py +344 -0
- autocoder/common/agents/integration_example.py +330 -0
- autocoder/common/agents/test_agent_parser.py +545 -0
- autocoder/common/async_utils.py +101 -0
- autocoder/common/auto_coder_lang.py +23 -972
- autocoder/common/autocoderargs_parser/__init__.py +14 -0
- autocoder/common/autocoderargs_parser/parser.py +184 -0
- autocoder/common/autocoderargs_parser/tests/__init__.py +1 -0
- autocoder/common/autocoderargs_parser/tests/test_args_parser.py +235 -0
- autocoder/common/autocoderargs_parser/tests/test_token_parser.py +195 -0
- autocoder/common/autocoderargs_parser/token_parser.py +290 -0
- autocoder/common/buildin_tokenizer.py +2 -4
- autocoder/common/code_auto_generate.py +149 -74
- autocoder/common/code_auto_generate_diff.py +163 -70
- autocoder/common/code_auto_generate_editblock.py +179 -89
- autocoder/common/code_auto_generate_strict_diff.py +167 -72
- autocoder/common/code_auto_merge_editblock.py +13 -6
- autocoder/common/code_modification_ranker.py +1 -1
- autocoder/common/command_completer.py +3 -3
- autocoder/common/command_file_manager/manager.py +183 -47
- autocoder/common/command_file_manager/test_command_file_manager.py +507 -0
- autocoder/common/command_templates.py +1 -1
- autocoder/common/conf_utils.py +2 -4
- autocoder/common/conversations/config.py +11 -3
- autocoder/common/conversations/get_conversation_manager.py +100 -2
- autocoder/common/conversations/llm_stats_models.py +264 -0
- autocoder/common/conversations/manager.py +112 -28
- autocoder/common/conversations/models.py +16 -2
- autocoder/common/conversations/storage/index_manager.py +134 -10
- autocoder/common/core_config/__init__.py +63 -0
- autocoder/common/core_config/agentic_mode_manager.py +109 -0
- autocoder/common/core_config/base_manager.py +123 -0
- autocoder/common/core_config/compatibility.py +151 -0
- autocoder/common/core_config/config_manager.py +156 -0
- autocoder/common/core_config/conversation_manager.py +31 -0
- autocoder/common/core_config/exclude_manager.py +72 -0
- autocoder/common/core_config/file_manager.py +177 -0
- autocoder/common/core_config/human_as_model_manager.py +129 -0
- autocoder/common/core_config/lib_manager.py +54 -0
- autocoder/common/core_config/main_manager.py +81 -0
- autocoder/common/core_config/mode_manager.py +126 -0
- autocoder/common/core_config/models.py +70 -0
- autocoder/common/core_config/test_memory_manager.py +1056 -0
- autocoder/common/env_manager.py +282 -0
- autocoder/common/env_manager_usage_example.py +211 -0
- autocoder/common/file_checkpoint/conversation_checkpoint.py +19 -19
- autocoder/common/file_checkpoint/manager.py +264 -48
- autocoder/common/file_checkpoint/test_backup.py +1 -18
- autocoder/common/file_checkpoint/test_manager.py +270 -1
- autocoder/common/file_checkpoint/test_store.py +1 -17
- autocoder/common/file_handler/__init__.py +23 -0
- autocoder/common/file_handler/active_context_handler.py +159 -0
- autocoder/common/file_handler/add_files_handler.py +409 -0
- autocoder/common/file_handler/chat_handler.py +180 -0
- autocoder/common/file_handler/coding_handler.py +401 -0
- autocoder/common/file_handler/commit_handler.py +200 -0
- autocoder/common/file_handler/lib_handler.py +156 -0
- autocoder/common/file_handler/list_files_handler.py +111 -0
- autocoder/common/file_handler/mcp_handler.py +268 -0
- autocoder/common/file_handler/models_handler.py +493 -0
- autocoder/common/file_handler/remove_files_handler.py +172 -0
- autocoder/common/file_monitor/test_file_monitor.py +307 -0
- autocoder/common/git_utils.py +51 -10
- autocoder/common/global_cancel.py +15 -6
- autocoder/common/ignorefiles/test_ignore_file_utils.py +1 -1
- autocoder/common/international/__init__.py +31 -0
- autocoder/common/international/demo_international.py +92 -0
- autocoder/common/international/message_manager.py +157 -0
- autocoder/common/international/messages/__init__.py +56 -0
- autocoder/common/international/messages/async_command_messages.py +507 -0
- autocoder/common/international/messages/auto_coder_messages.py +2208 -0
- autocoder/common/international/messages/chat_auto_coder_messages.py +1547 -0
- autocoder/common/international/messages/command_help_messages.py +986 -0
- autocoder/common/international/messages/conversation_command_messages.py +191 -0
- autocoder/common/international/messages/git_helper_plugin_messages.py +159 -0
- autocoder/common/international/messages/queue_command_messages.py +751 -0
- autocoder/common/international/messages/rules_command_messages.py +77 -0
- autocoder/common/international/messages/sdk_messages.py +1707 -0
- autocoder/common/international/messages/token_helper_plugin_messages.py +361 -0
- autocoder/common/international/messages/tool_display_messages.py +1212 -0
- autocoder/common/international/messages/workflow_exception_messages.py +473 -0
- autocoder/common/international/test_international.py +612 -0
- autocoder/common/linter_core/__init__.py +28 -0
- autocoder/common/linter_core/base_linter.py +61 -0
- autocoder/common/linter_core/config_loader.py +271 -0
- autocoder/common/linter_core/formatters/__init__.py +0 -0
- autocoder/common/linter_core/formatters/base_formatter.py +38 -0
- autocoder/common/linter_core/formatters/raw_formatter.py +17 -0
- autocoder/common/linter_core/linter.py +166 -0
- autocoder/common/linter_core/linter_factory.py +216 -0
- autocoder/common/linter_core/linter_manager.py +333 -0
- autocoder/common/linter_core/linters/__init__.py +9 -0
- autocoder/common/linter_core/linters/java_linter.py +342 -0
- autocoder/common/linter_core/linters/python_linter.py +115 -0
- autocoder/common/linter_core/linters/typescript_linter.py +119 -0
- autocoder/common/linter_core/models/__init__.py +7 -0
- autocoder/common/linter_core/models/lint_result.py +91 -0
- autocoder/common/linter_core/models.py +33 -0
- autocoder/common/linter_core/tests/__init__.py +3 -0
- autocoder/common/linter_core/tests/test_config_loader.py +323 -0
- autocoder/common/linter_core/tests/test_config_loading.py +308 -0
- autocoder/common/linter_core/tests/test_factory_manager.py +234 -0
- autocoder/common/linter_core/tests/test_formatters.py +147 -0
- autocoder/common/linter_core/tests/test_integration.py +317 -0
- autocoder/common/linter_core/tests/test_java_linter.py +496 -0
- autocoder/common/linter_core/tests/test_linters.py +265 -0
- autocoder/common/linter_core/tests/test_models.py +81 -0
- autocoder/common/linter_core/tests/verify_config_loading.py +296 -0
- autocoder/common/linter_core/tests/verify_fixes.py +183 -0
- autocoder/common/llm_friendly_package/__init__.py +31 -0
- autocoder/common/llm_friendly_package/base_manager.py +102 -0
- autocoder/common/llm_friendly_package/docs_manager.py +121 -0
- autocoder/common/llm_friendly_package/library_manager.py +171 -0
- autocoder/common/{llm_friendly_package.py → llm_friendly_package/main_manager.py} +204 -231
- autocoder/common/llm_friendly_package/models.py +40 -0
- autocoder/common/llm_friendly_package/test_llm_friendly_package.py +536 -0
- autocoder/common/llms/__init__.py +15 -0
- autocoder/common/llms/demo_error_handling.py +85 -0
- autocoder/common/llms/factory.py +142 -0
- autocoder/common/llms/manager.py +264 -0
- autocoder/common/llms/pricing.py +121 -0
- autocoder/common/llms/registry.py +288 -0
- autocoder/common/llms/schema.py +77 -0
- autocoder/common/llms/simple_demo.py +45 -0
- autocoder/common/llms/test_quick_model.py +116 -0
- autocoder/common/llms/test_remove_functionality.py +182 -0
- autocoder/common/llms/tests/__init__.py +1 -0
- autocoder/common/llms/tests/test_manager.py +330 -0
- autocoder/common/llms/tests/test_registry.py +364 -0
- autocoder/common/mcp_tools/__init__.py +62 -0
- autocoder/common/{mcp_tools.py → mcp_tools/executor.py} +49 -40
- autocoder/common/{mcp_hub.py → mcp_tools/hub.py} +42 -68
- autocoder/common/{mcp_server_install.py → mcp_tools/installer.py} +16 -28
- autocoder/common/{mcp_server.py → mcp_tools/server.py} +176 -48
- autocoder/common/mcp_tools/test_keyboard_interrupt.py +93 -0
- autocoder/common/mcp_tools/test_mcp_tools.py +391 -0
- autocoder/common/{mcp_server_types.py → mcp_tools/types.py} +121 -48
- autocoder/common/mcp_tools/verify_functionality.py +202 -0
- autocoder/common/model_speed_tester.py +32 -26
- autocoder/common/priority_directory_finder/__init__.py +142 -0
- autocoder/common/priority_directory_finder/examples.py +230 -0
- autocoder/common/priority_directory_finder/finder.py +283 -0
- autocoder/common/priority_directory_finder/models.py +236 -0
- autocoder/common/priority_directory_finder/test_priority_directory_finder.py +431 -0
- autocoder/common/project_scanner/__init__.py +18 -0
- autocoder/common/project_scanner/compat.py +77 -0
- autocoder/common/project_scanner/scanner.py +436 -0
- autocoder/common/project_tracker/__init__.py +27 -0
- autocoder/common/project_tracker/api.py +228 -0
- autocoder/common/project_tracker/demo.py +272 -0
- autocoder/common/project_tracker/tracker.py +487 -0
- autocoder/common/project_tracker/types.py +53 -0
- autocoder/common/pruner/__init__.py +67 -0
- autocoder/common/pruner/agentic_conversation_pruner.py +746 -0
- autocoder/common/{context_pruner.py → pruner/context_pruner.py} +137 -40
- autocoder/common/pruner/conversation_message_ids_api.py +386 -0
- autocoder/common/pruner/conversation_message_ids_manager.py +347 -0
- autocoder/common/pruner/conversation_message_ids_pruner.py +473 -0
- autocoder/common/pruner/conversation_normalizer.py +347 -0
- autocoder/common/{conversation_pruner.py → pruner/conversation_pruner.py} +26 -6
- autocoder/common/pruner/test_agentic_conversation_pruner.py +784 -0
- autocoder/common/pruner/test_context_pruner.py +546 -0
- autocoder/common/pruner/test_conversation_normalizer.py +502 -0
- autocoder/common/pruner/test_tool_content_detector.py +324 -0
- autocoder/common/pruner/tool_content_detector.py +227 -0
- autocoder/common/pruner/tools/__init__.py +18 -0
- autocoder/common/pruner/tools/query_message_ids.py +264 -0
- autocoder/common/pruner/tools/test_agentic_pruning_logic.py +432 -0
- autocoder/common/pruner/tools/test_message_ids_pruning_only.py +192 -0
- autocoder/common/pull_requests/__init__.py +9 -1
- autocoder/common/pull_requests/utils.py +122 -1
- autocoder/common/rag_manager/rag_manager.py +36 -40
- autocoder/common/rulefiles/__init__.py +53 -1
- autocoder/common/rulefiles/api.py +250 -0
- autocoder/common/rulefiles/core/__init__.py +14 -0
- autocoder/common/rulefiles/core/manager.py +241 -0
- autocoder/common/rulefiles/core/selector.py +805 -0
- autocoder/common/rulefiles/models/__init__.py +20 -0
- autocoder/common/rulefiles/models/index.py +16 -0
- autocoder/common/rulefiles/models/init_rule.py +18 -0
- autocoder/common/rulefiles/models/rule_file.py +18 -0
- autocoder/common/rulefiles/models/rule_relevance.py +14 -0
- autocoder/common/rulefiles/models/summary.py +16 -0
- autocoder/common/rulefiles/test_rulefiles.py +776 -0
- autocoder/common/rulefiles/utils/__init__.py +34 -0
- autocoder/common/rulefiles/utils/monitor.py +86 -0
- autocoder/common/rulefiles/utils/parser.py +230 -0
- autocoder/common/save_formatted_log.py +67 -10
- autocoder/common/search_replace.py +8 -1
- autocoder/common/search_replace_patch/__init__.py +24 -0
- autocoder/common/search_replace_patch/base.py +115 -0
- autocoder/common/search_replace_patch/manager.py +248 -0
- autocoder/common/search_replace_patch/patch_replacer.py +304 -0
- autocoder/common/search_replace_patch/similarity_replacer.py +306 -0
- autocoder/common/search_replace_patch/string_replacer.py +181 -0
- autocoder/common/search_replace_patch/tests/__init__.py +3 -0
- autocoder/common/search_replace_patch/tests/run_tests.py +126 -0
- autocoder/common/search_replace_patch/tests/test_base.py +188 -0
- autocoder/common/search_replace_patch/tests/test_empty_line_insert.py +233 -0
- autocoder/common/search_replace_patch/tests/test_integration.py +389 -0
- autocoder/common/search_replace_patch/tests/test_manager.py +351 -0
- autocoder/common/search_replace_patch/tests/test_patch_replacer.py +316 -0
- autocoder/common/search_replace_patch/tests/test_regex_replacer.py +306 -0
- autocoder/common/search_replace_patch/tests/test_similarity_replacer.py +384 -0
- autocoder/common/shell_commands/__init__.py +197 -0
- autocoder/common/shell_commands/background_process_notifier.py +346 -0
- autocoder/common/shell_commands/command_executor.py +1127 -0
- autocoder/common/shell_commands/error_recovery.py +541 -0
- autocoder/common/shell_commands/exceptions.py +120 -0
- autocoder/common/shell_commands/interactive_executor.py +476 -0
- autocoder/common/shell_commands/interactive_pexpect_process.py +623 -0
- autocoder/common/shell_commands/interactive_process.py +744 -0
- autocoder/common/shell_commands/interactive_session_manager.py +1014 -0
- autocoder/common/shell_commands/monitoring.py +529 -0
- autocoder/common/shell_commands/process_cleanup.py +386 -0
- autocoder/common/shell_commands/process_manager.py +606 -0
- autocoder/common/shell_commands/test_interactive_pexpect_process.py +281 -0
- autocoder/common/shell_commands/tests/__init__.py +6 -0
- autocoder/common/shell_commands/tests/conftest.py +118 -0
- autocoder/common/shell_commands/tests/test_background_process_notifier.py +703 -0
- autocoder/common/shell_commands/tests/test_command_executor.py +448 -0
- autocoder/common/shell_commands/tests/test_error_recovery.py +305 -0
- autocoder/common/shell_commands/tests/test_exceptions.py +299 -0
- autocoder/common/shell_commands/tests/test_execute_batch.py +588 -0
- autocoder/common/shell_commands/tests/test_indented_batch_commands.py +244 -0
- autocoder/common/shell_commands/tests/test_integration.py +664 -0
- autocoder/common/shell_commands/tests/test_monitoring.py +546 -0
- autocoder/common/shell_commands/tests/test_performance.py +632 -0
- autocoder/common/shell_commands/tests/test_process_cleanup.py +397 -0
- autocoder/common/shell_commands/tests/test_process_manager.py +606 -0
- autocoder/common/shell_commands/tests/test_timeout_config.py +343 -0
- autocoder/common/shell_commands/tests/test_timeout_manager.py +520 -0
- autocoder/common/shell_commands/timeout_config.py +315 -0
- autocoder/common/shell_commands/timeout_manager.py +352 -0
- autocoder/common/terminal_paste/__init__.py +14 -0
- autocoder/common/terminal_paste/demo.py +145 -0
- autocoder/common/terminal_paste/demo_paste_functionality.py +95 -0
- autocoder/common/terminal_paste/paste_handler.py +200 -0
- autocoder/common/terminal_paste/paste_manager.py +118 -0
- autocoder/common/terminal_paste/tests/__init__.py +1 -0
- autocoder/common/terminal_paste/tests/test_paste_handler.py +182 -0
- autocoder/common/terminal_paste/tests/test_paste_manager.py +126 -0
- autocoder/common/terminal_paste/utils.py +163 -0
- autocoder/common/test_autocoder_args.py +232 -0
- autocoder/common/test_env_manager.py +173 -0
- autocoder/common/test_env_manager_integration.py +159 -0
- autocoder/common/text_similarity/__init__.py +9 -0
- autocoder/common/text_similarity/demo.py +216 -0
- autocoder/common/text_similarity/examples.py +266 -0
- autocoder/common/text_similarity/test_text_similarity.py +306 -0
- autocoder/common/text_similarity/text_similarity.py +194 -0
- autocoder/common/text_similarity/utils.py +125 -0
- autocoder/common/todos/__init__.py +61 -0
- autocoder/common/todos/cache/__init__.py +16 -0
- autocoder/common/todos/cache/base_cache.py +89 -0
- autocoder/common/todos/cache/cache_manager.py +228 -0
- autocoder/common/todos/cache/memory_cache.py +225 -0
- autocoder/common/todos/config.py +155 -0
- autocoder/common/todos/exceptions.py +35 -0
- autocoder/common/todos/get_todo_manager.py +161 -0
- autocoder/common/todos/manager.py +537 -0
- autocoder/common/todos/models.py +239 -0
- autocoder/common/todos/storage/__init__.py +14 -0
- autocoder/common/todos/storage/base_storage.py +76 -0
- autocoder/common/todos/storage/file_storage.py +278 -0
- autocoder/common/tokens/__init__.py +15 -0
- autocoder/common/tokens/counter.py +44 -2
- autocoder/common/tools_manager/__init__.py +17 -0
- autocoder/common/tools_manager/examples.py +162 -0
- autocoder/common/tools_manager/manager.py +385 -0
- autocoder/common/tools_manager/models.py +39 -0
- autocoder/common/tools_manager/test_tools_manager.py +303 -0
- autocoder/common/tools_manager/utils.py +191 -0
- autocoder/common/v2/agent/agentic_callbacks.py +270 -0
- autocoder/common/v2/agent/agentic_edit.py +2729 -2052
- autocoder/common/v2/agent/agentic_edit_change_manager.py +474 -0
- autocoder/common/v2/agent/agentic_edit_tools/__init__.py +43 -2
- autocoder/common/v2/agent/agentic_edit_tools/ac_mod_list_tool_resolver.py +279 -0
- autocoder/common/v2/agent/agentic_edit_tools/ac_mod_read_tool_resolver.py +40 -0
- autocoder/common/v2/agent/agentic_edit_tools/ac_mod_write_tool_resolver.py +52 -0
- autocoder/common/v2/agent/agentic_edit_tools/ask_followup_question_tool_resolver.py +8 -0
- autocoder/common/v2/agent/agentic_edit_tools/background_task_tool_resolver.py +1167 -0
- autocoder/common/v2/agent/agentic_edit_tools/base_tool_resolver.py +2 -2
- autocoder/common/v2/agent/agentic_edit_tools/conversation_message_ids_read_tool_resolver.py +214 -0
- autocoder/common/v2/agent/agentic_edit_tools/conversation_message_ids_write_tool_resolver.py +299 -0
- autocoder/common/v2/agent/agentic_edit_tools/count_tokens_tool_resolver.py +290 -0
- autocoder/common/v2/agent/agentic_edit_tools/execute_command_tool_resolver.py +565 -30
- autocoder/common/v2/agent/agentic_edit_tools/execute_workflow_tool_resolver.py +485 -0
- autocoder/common/v2/agent/agentic_edit_tools/extract_to_text_tool_resolver.py +225 -0
- autocoder/common/v2/agent/agentic_edit_tools/lint_report.py +79 -0
- autocoder/common/v2/agent/agentic_edit_tools/linter_config_models.py +343 -0
- autocoder/common/v2/agent/agentic_edit_tools/linter_enabled_tool_resolver.py +189 -0
- autocoder/common/v2/agent/agentic_edit_tools/list_files_tool_resolver.py +169 -101
- autocoder/common/v2/agent/agentic_edit_tools/load_extra_document_tool_resolver.py +349 -0
- autocoder/common/v2/agent/agentic_edit_tools/read_file_tool_resolver.py +244 -51
- autocoder/common/v2/agent/agentic_edit_tools/replace_in_file_tool_resolver.py +667 -147
- autocoder/common/v2/agent/agentic_edit_tools/run_named_subagents_tool_resolver.py +691 -0
- autocoder/common/v2/agent/agentic_edit_tools/search_files_tool_resolver.py +409 -140
- autocoder/common/v2/agent/agentic_edit_tools/session_interactive_tool_resolver.py +115 -0
- autocoder/common/v2/agent/agentic_edit_tools/session_start_tool_resolver.py +190 -0
- autocoder/common/v2/agent/agentic_edit_tools/session_stop_tool_resolver.py +76 -0
- autocoder/common/v2/agent/agentic_edit_tools/test_write_to_file_tool_resolver.py +209 -194
- autocoder/common/v2/agent/agentic_edit_tools/todo_read_tool_resolver.py +135 -0
- autocoder/common/v2/agent/agentic_edit_tools/todo_write_tool_resolver.py +328 -0
- autocoder/common/v2/agent/agentic_edit_tools/use_mcp_tool_resolver.py +2 -2
- autocoder/common/v2/agent/agentic_edit_tools/web_crawl_tool_resolver.py +557 -0
- autocoder/common/v2/agent/agentic_edit_tools/web_search_tool_resolver.py +600 -0
- autocoder/common/v2/agent/agentic_edit_tools/write_to_file_tool_resolver.py +56 -121
- autocoder/common/v2/agent/agentic_edit_types.py +386 -10
- autocoder/common/v2/agent/runner/__init__.py +31 -0
- autocoder/common/v2/agent/runner/base_runner.py +92 -0
- autocoder/common/v2/agent/runner/file_based_event_runner.py +217 -0
- autocoder/common/v2/agent/runner/sdk_runner.py +182 -0
- autocoder/common/v2/agent/runner/terminal_runner.py +396 -0
- autocoder/common/v2/agent/runner/tool_display.py +589 -0
- autocoder/common/v2/agent/test_agentic_callbacks.py +265 -0
- autocoder/common/v2/agent/test_agentic_edit.py +194 -0
- autocoder/common/v2/agent/tool_caller/__init__.py +24 -0
- autocoder/common/v2/agent/tool_caller/default_tool_resolver_map.py +135 -0
- autocoder/common/v2/agent/tool_caller/integration_test.py +172 -0
- autocoder/common/v2/agent/tool_caller/plugins/__init__.py +14 -0
- autocoder/common/v2/agent/tool_caller/plugins/base_plugin.py +126 -0
- autocoder/common/v2/agent/tool_caller/plugins/examples/__init__.py +13 -0
- autocoder/common/v2/agent/tool_caller/plugins/examples/logging_plugin.py +164 -0
- autocoder/common/v2/agent/tool_caller/plugins/examples/security_filter_plugin.py +198 -0
- autocoder/common/v2/agent/tool_caller/plugins/plugin_interface.py +141 -0
- autocoder/common/v2/agent/tool_caller/test_tool_caller.py +278 -0
- autocoder/common/v2/agent/tool_caller/tool_call_plugin_manager.py +331 -0
- autocoder/common/v2/agent/tool_caller/tool_caller.py +337 -0
- autocoder/common/v2/agent/tool_caller/usage_example.py +193 -0
- autocoder/common/v2/code_agentic_editblock_manager.py +4 -4
- autocoder/common/v2/code_auto_generate.py +136 -78
- autocoder/common/v2/code_auto_generate_diff.py +135 -79
- autocoder/common/v2/code_auto_generate_editblock.py +174 -99
- autocoder/common/v2/code_auto_generate_strict_diff.py +151 -71
- autocoder/common/v2/code_auto_merge.py +1 -1
- autocoder/common/v2/code_auto_merge_editblock.py +13 -1
- autocoder/common/v2/code_diff_manager.py +3 -3
- autocoder/common/v2/code_editblock_manager.py +4 -14
- autocoder/common/v2/code_manager.py +1 -1
- autocoder/common/v2/code_strict_diff_manager.py +2 -2
- autocoder/common/wrap_llm_hint/__init__.py +10 -0
- autocoder/common/wrap_llm_hint/test_wrap_llm_hint.py +1067 -0
- autocoder/common/wrap_llm_hint/utils.py +432 -0
- autocoder/common/wrap_llm_hint/wrap_llm_hint.py +323 -0
- autocoder/completer/__init__.py +8 -0
- autocoder/completer/command_completer_v2.py +1051 -0
- autocoder/default_project/__init__.py +501 -0
- autocoder/dispacher/__init__.py +4 -12
- autocoder/dispacher/actions/action.py +165 -7
- autocoder/dispacher/actions/plugins/action_regex_project.py +2 -2
- autocoder/index/entry.py +117 -125
- autocoder/{agent → index/filter}/agentic_filter.py +323 -334
- autocoder/index/filter/normal_filter.py +5 -11
- autocoder/index/filter/quick_filter.py +1 -1
- autocoder/index/index.py +36 -9
- autocoder/index/tests/__init__.py +1 -0
- autocoder/index/tests/run_tests.py +195 -0
- autocoder/index/tests/test_entry.py +303 -0
- autocoder/index/tests/test_index_manager.py +314 -0
- autocoder/index/tests/test_module_integration.py +300 -0
- autocoder/index/tests/test_symbols_utils.py +183 -0
- autocoder/inner/__init__.py +4 -0
- autocoder/inner/agentic.py +932 -0
- autocoder/inner/async_command_handler.py +992 -0
- autocoder/inner/conversation_command_handlers.py +623 -0
- autocoder/inner/merge_command_handler.py +213 -0
- autocoder/inner/queue_command_handler.py +684 -0
- autocoder/models.py +95 -266
- autocoder/plugins/git_helper_plugin.py +31 -29
- autocoder/plugins/token_helper_plugin.py +156 -37
- autocoder/pyproject/__init__.py +32 -29
- autocoder/rag/agentic_rag.py +215 -75
- autocoder/rag/cache/simple_cache.py +1 -2
- autocoder/rag/loaders/image_loader.py +1 -1
- autocoder/rag/long_context_rag.py +42 -26
- autocoder/rag/qa_conversation_strategy.py +1 -1
- autocoder/rag/terminal/__init__.py +17 -0
- autocoder/rag/terminal/args.py +581 -0
- autocoder/rag/terminal/bootstrap.py +61 -0
- autocoder/rag/terminal/command_handlers.py +653 -0
- autocoder/rag/terminal/formatters/__init__.py +20 -0
- autocoder/rag/terminal/formatters/base.py +70 -0
- autocoder/rag/terminal/formatters/json_format.py +66 -0
- autocoder/rag/terminal/formatters/stream_json.py +95 -0
- autocoder/rag/terminal/formatters/text.py +28 -0
- autocoder/rag/terminal/init.py +120 -0
- autocoder/rag/terminal/utils.py +106 -0
- autocoder/rag/test_agentic_rag.py +389 -0
- autocoder/rag/test_doc_filter.py +3 -3
- autocoder/rag/test_long_context_rag.py +1 -1
- autocoder/rag/test_token_limiter.py +517 -10
- autocoder/rag/token_counter.py +3 -0
- autocoder/rag/token_limiter.py +19 -15
- autocoder/rag/tools/__init__.py +26 -2
- autocoder/rag/tools/bochaai_example.py +343 -0
- autocoder/rag/tools/bochaai_sdk.py +541 -0
- autocoder/rag/tools/metaso_example.py +268 -0
- autocoder/rag/tools/metaso_sdk.py +417 -0
- autocoder/rag/tools/recall_tool.py +28 -7
- autocoder/rag/tools/run_integration_tests.py +204 -0
- autocoder/rag/tools/test_all_providers.py +318 -0
- autocoder/rag/tools/test_bochaai_integration.py +482 -0
- autocoder/rag/tools/test_final_integration.py +215 -0
- autocoder/rag/tools/test_metaso_integration.py +424 -0
- autocoder/rag/tools/test_metaso_real.py +171 -0
- autocoder/rag/tools/test_web_crawl_tool.py +639 -0
- autocoder/rag/tools/test_web_search_tool.py +509 -0
- autocoder/rag/tools/todo_read_tool.py +202 -0
- autocoder/rag/tools/todo_write_tool.py +412 -0
- autocoder/rag/tools/web_crawl_tool.py +634 -0
- autocoder/rag/tools/web_search_tool.py +558 -0
- autocoder/rag/tools/web_tools_example.py +119 -0
- autocoder/rag/types.py +16 -0
- autocoder/rag/variable_holder.py +4 -2
- autocoder/rags.py +86 -79
- autocoder/regexproject/__init__.py +23 -21
- autocoder/run_context.py +9 -0
- autocoder/sdk/__init__.py +50 -161
- autocoder/sdk/api.py +370 -0
- autocoder/sdk/async_runner/__init__.py +26 -0
- autocoder/sdk/async_runner/async_executor.py +650 -0
- autocoder/sdk/async_runner/async_handler.py +356 -0
- autocoder/sdk/async_runner/markdown_processor.py +595 -0
- autocoder/sdk/async_runner/task_metadata.py +284 -0
- autocoder/sdk/async_runner/worktree_manager.py +438 -0
- autocoder/sdk/cli/__init__.py +2 -5
- autocoder/sdk/cli/formatters.py +28 -204
- autocoder/sdk/cli/handlers.py +77 -44
- autocoder/sdk/cli/main.py +158 -170
- autocoder/sdk/cli/options.py +95 -22
- autocoder/sdk/constants.py +139 -51
- autocoder/sdk/core/auto_coder_core.py +484 -267
- autocoder/sdk/core/bridge.py +298 -118
- autocoder/sdk/exceptions.py +18 -12
- autocoder/sdk/formatters/__init__.py +19 -0
- autocoder/sdk/formatters/input.py +64 -0
- autocoder/sdk/formatters/output.py +247 -0
- autocoder/sdk/formatters/stream.py +54 -0
- autocoder/sdk/models/__init__.py +6 -5
- autocoder/sdk/models/options.py +55 -18
- autocoder/sdk/utils/formatters.py +27 -195
- autocoder/suffixproject/__init__.py +28 -25
- autocoder/terminal/__init__.py +14 -0
- autocoder/terminal/app.py +454 -0
- autocoder/terminal/args.py +32 -0
- autocoder/terminal/bootstrap.py +178 -0
- autocoder/terminal/command_processor.py +521 -0
- autocoder/terminal/command_registry.py +57 -0
- autocoder/terminal/help.py +97 -0
- autocoder/terminal/tasks/__init__.py +5 -0
- autocoder/terminal/tasks/background.py +77 -0
- autocoder/terminal/tasks/task_event.py +70 -0
- autocoder/terminal/ui/__init__.py +13 -0
- autocoder/terminal/ui/completer.py +268 -0
- autocoder/terminal/ui/keybindings.py +75 -0
- autocoder/terminal/ui/session.py +41 -0
- autocoder/terminal/ui/toolbar.py +64 -0
- autocoder/terminal/utils/__init__.py +13 -0
- autocoder/terminal/utils/errors.py +18 -0
- autocoder/terminal/utils/paths.py +19 -0
- autocoder/terminal/utils/shell.py +43 -0
- autocoder/terminal_v3/__init__.py +10 -0
- autocoder/terminal_v3/app.py +201 -0
- autocoder/terminal_v3/handlers/__init__.py +5 -0
- autocoder/terminal_v3/handlers/command_handler.py +131 -0
- autocoder/terminal_v3/models/__init__.py +6 -0
- autocoder/terminal_v3/models/conversation_buffer.py +214 -0
- autocoder/terminal_v3/models/message.py +50 -0
- autocoder/terminal_v3/models/tool_display.py +247 -0
- autocoder/terminal_v3/ui/__init__.py +7 -0
- autocoder/terminal_v3/ui/keybindings.py +56 -0
- autocoder/terminal_v3/ui/layout.py +141 -0
- autocoder/terminal_v3/ui/styles.py +43 -0
- autocoder/tsproject/__init__.py +23 -23
- autocoder/utils/auto_coder_utils/chat_stream_out.py +1 -1
- autocoder/utils/llms.py +88 -80
- autocoder/utils/math_utils.py +101 -0
- autocoder/utils/model_provider_selector.py +16 -4
- autocoder/utils/operate_config_api.py +33 -5
- autocoder/utils/thread_utils.py +2 -2
- autocoder/version.py +4 -2
- autocoder/workflow_agents/__init__.py +84 -0
- autocoder/workflow_agents/agent.py +143 -0
- autocoder/workflow_agents/exceptions.py +573 -0
- autocoder/workflow_agents/executor.py +489 -0
- autocoder/workflow_agents/loader.py +737 -0
- autocoder/workflow_agents/runner.py +267 -0
- autocoder/workflow_agents/types.py +172 -0
- autocoder/workflow_agents/utils.py +434 -0
- autocoder/workflow_agents/workflow_manager.py +211 -0
- auto_coder-0.1.400.dist-info/METADATA +0 -396
- auto_coder-0.1.400.dist-info/RECORD +0 -425
- auto_coder-0.1.400.dist-info/licenses/LICENSE +0 -201
- autocoder/auto_coder_server.py +0 -672
- autocoder/benchmark.py +0 -138
- autocoder/common/ac_style_command_parser/example.py +0 -7
- autocoder/common/cleaner.py +0 -31
- autocoder/common/command_completer_v2.py +0 -615
- autocoder/common/directory_cache/__init__.py +0 -1
- autocoder/common/directory_cache/cache.py +0 -192
- autocoder/common/directory_cache/test_cache.py +0 -190
- autocoder/common/file_checkpoint/examples.py +0 -217
- autocoder/common/llm_friendly_package_example.py +0 -138
- autocoder/common/llm_friendly_package_test.py +0 -63
- autocoder/common/pull_requests/test_module.py +0 -1
- autocoder/common/rulefiles/autocoderrules_utils.py +0 -484
- autocoder/common/text.py +0 -30
- autocoder/common/v2/agent/agentic_edit_tools/list_package_info_tool_resolver.py +0 -42
- autocoder/common/v2/agent/agentic_edit_tools/test_execute_command_tool_resolver.py +0 -70
- autocoder/common/v2/agent/agentic_edit_tools/test_search_files_tool_resolver.py +0 -163
- autocoder/common/v2/agent/agentic_tool_display.py +0 -183
- autocoder/plugins/dynamic_completion_example.py +0 -148
- autocoder/plugins/sample_plugin.py +0 -160
- autocoder/sdk/cli/__main__.py +0 -26
- autocoder/sdk/cli/completion_wrapper.py +0 -38
- autocoder/sdk/cli/install_completion.py +0 -301
- autocoder/sdk/models/messages.py +0 -209
- autocoder/sdk/session/__init__.py +0 -32
- autocoder/sdk/session/session.py +0 -106
- autocoder/sdk/session/session_manager.py +0 -56
- {auto_coder-0.1.400.dist-info → auto_coder-2.0.0.dist-info}/top_level.txt +0 -0
- /autocoder/{sdk/example.py → common/agent_query_queue/__init__.py} +0 -0
|
@@ -0,0 +1,784 @@
|
|
|
1
|
+
"""
|
|
2
|
+
Pytest tests for AgenticConversationPruner
|
|
3
|
+
|
|
4
|
+
This module contains comprehensive tests for the AgenticConversationPruner class,
|
|
5
|
+
including functionality tests, edge cases, and integration tests.
|
|
6
|
+
"""
|
|
7
|
+
|
|
8
|
+
import json
|
|
9
|
+
import pytest
|
|
10
|
+
import tempfile
|
|
11
|
+
import shutil
|
|
12
|
+
import os
|
|
13
|
+
from unittest.mock import MagicMock, patch
|
|
14
|
+
from autocoder.common.pruner.agentic_conversation_pruner import AgenticConversationPruner
|
|
15
|
+
from autocoder.common import AutoCoderArgs
|
|
16
|
+
from autocoder.sdk import get_llm, init_project_if_required
|
|
17
|
+
from autocoder.common.conf_utils import load_tokenizer
|
|
18
|
+
from autocoder.common.tokens import count_string_tokens
|
|
19
|
+
|
|
20
|
+
load_tokenizer()
|
|
21
|
+
|
|
22
|
+
class TestAgenticConversationPruner:
|
|
23
|
+
"""Test suite for AgenticConversationPruner class"""
|
|
24
|
+
|
|
25
|
+
@pytest.fixture
|
|
26
|
+
def temp_test_dir(self):
|
|
27
|
+
"""提供一个临时的、测试后自动清理的目录"""
|
|
28
|
+
# 保存原始工作目录
|
|
29
|
+
original_cwd = os.getcwd()
|
|
30
|
+
temp_dir = tempfile.mkdtemp()
|
|
31
|
+
try:
|
|
32
|
+
yield temp_dir
|
|
33
|
+
finally:
|
|
34
|
+
# 确保恢复到原始目录,即使出现异常
|
|
35
|
+
try:
|
|
36
|
+
os.chdir(original_cwd)
|
|
37
|
+
except OSError:
|
|
38
|
+
# 如果原始目录也不存在,则切换到用户主目录
|
|
39
|
+
os.chdir(os.path.expanduser("~"))
|
|
40
|
+
# 删除临时目录
|
|
41
|
+
if os.path.exists(temp_dir):
|
|
42
|
+
shutil.rmtree(temp_dir)
|
|
43
|
+
|
|
44
|
+
@pytest.fixture
|
|
45
|
+
def mock_args(self):
|
|
46
|
+
"""Create mock AutoCoderArgs for testing"""
|
|
47
|
+
return AutoCoderArgs(
|
|
48
|
+
source_dir=".",
|
|
49
|
+
conversation_prune_safe_zone_tokens=1000, # Small threshold for testing
|
|
50
|
+
context_prune=True,
|
|
51
|
+
context_prune_strategy="extract",
|
|
52
|
+
context_prune_sliding_window_size=10,
|
|
53
|
+
context_prune_sliding_window_overlap=2,
|
|
54
|
+
query="请帮我分析这些代码"
|
|
55
|
+
)
|
|
56
|
+
|
|
57
|
+
@pytest.fixture
|
|
58
|
+
def mock_args_small_threshold(self):
|
|
59
|
+
"""Create mock AutoCoderArgs with very small token threshold for testing"""
|
|
60
|
+
return AutoCoderArgs(
|
|
61
|
+
source_dir=".",
|
|
62
|
+
conversation_prune_safe_zone_tokens=50, # Very small threshold to trigger pruning
|
|
63
|
+
context_prune=True,
|
|
64
|
+
context_prune_strategy="extract",
|
|
65
|
+
context_prune_sliding_window_size=10,
|
|
66
|
+
context_prune_sliding_window_overlap=2,
|
|
67
|
+
query="请帮我分析这些代码"
|
|
68
|
+
)
|
|
69
|
+
|
|
70
|
+
@pytest.fixture
|
|
71
|
+
def real_llm(self):
|
|
72
|
+
"""创建真实的LLM对象"""
|
|
73
|
+
llm = get_llm("v3_chat", product_mode="lite")
|
|
74
|
+
return llm
|
|
75
|
+
|
|
76
|
+
|
|
77
|
+
|
|
78
|
+
@pytest.fixture
|
|
79
|
+
def pruner(self, mock_args, real_llm):
|
|
80
|
+
"""Create AgenticConversationPruner instance for testing"""
|
|
81
|
+
return AgenticConversationPruner(args=mock_args, llm=real_llm, conversation_id="test-conversation-id")
|
|
82
|
+
|
|
83
|
+
@pytest.fixture
|
|
84
|
+
def sample_file_sources(self, temp_test_dir):
|
|
85
|
+
"""Sample file sources for testing
|
|
86
|
+
Creates a simulated project structure in the temporary directory
|
|
87
|
+
"""
|
|
88
|
+
# 创建项目结构
|
|
89
|
+
src_dir = os.path.join(temp_test_dir, "src")
|
|
90
|
+
utils_dir = os.path.join(src_dir, "utils")
|
|
91
|
+
os.makedirs(utils_dir, exist_ok=True)
|
|
92
|
+
|
|
93
|
+
# 创建 __init__.py 文件使其成为有效的 Python 包
|
|
94
|
+
with open(os.path.join(src_dir, "__init__.py"), "w") as f:
|
|
95
|
+
f.write("# src package")
|
|
96
|
+
with open(os.path.join(utils_dir, "__init__.py"), "w") as f:
|
|
97
|
+
f.write("# utils package")
|
|
98
|
+
|
|
99
|
+
# 创建数学工具模块
|
|
100
|
+
math_utils_content = '''def add(a, b):
|
|
101
|
+
"""加法函数"""
|
|
102
|
+
return a + b
|
|
103
|
+
|
|
104
|
+
def subtract(a, b):
|
|
105
|
+
"""减法函数"""
|
|
106
|
+
return a - b
|
|
107
|
+
|
|
108
|
+
def multiply(a, b):
|
|
109
|
+
"""乘法函数"""
|
|
110
|
+
return a * b
|
|
111
|
+
|
|
112
|
+
def divide(a, b):
|
|
113
|
+
"""除法函数"""
|
|
114
|
+
if b == 0:
|
|
115
|
+
raise ValueError("Cannot divide by zero")
|
|
116
|
+
return a / b
|
|
117
|
+
'''
|
|
118
|
+
math_utils_path = os.path.join(utils_dir, "math_utils.py")
|
|
119
|
+
with open(math_utils_path, "w") as f:
|
|
120
|
+
f.write(math_utils_content)
|
|
121
|
+
|
|
122
|
+
# 创建字符串工具模块
|
|
123
|
+
string_utils_content = '''def format_string(s):
|
|
124
|
+
"""格式化字符串"""
|
|
125
|
+
return s.strip().lower()
|
|
126
|
+
|
|
127
|
+
def reverse_string(s):
|
|
128
|
+
"""反转字符串"""
|
|
129
|
+
return s[::-1]
|
|
130
|
+
|
|
131
|
+
def count_characters(s):
|
|
132
|
+
"""计算字符数"""
|
|
133
|
+
return len(s)
|
|
134
|
+
'''
|
|
135
|
+
string_utils_path = os.path.join(utils_dir, "string_utils.py")
|
|
136
|
+
with open(string_utils_path, "w") as f:
|
|
137
|
+
f.write(string_utils_content)
|
|
138
|
+
|
|
139
|
+
# 创建主程序文件
|
|
140
|
+
main_content = '''from utils.math_utils import add, subtract
|
|
141
|
+
from utils.string_utils import format_string
|
|
142
|
+
|
|
143
|
+
def main():
|
|
144
|
+
print("计算结果:", add(5, 3))
|
|
145
|
+
print("格式化结果:", format_string(" Hello World "))
|
|
146
|
+
|
|
147
|
+
if __name__ == "__main__":
|
|
148
|
+
main()
|
|
149
|
+
'''
|
|
150
|
+
main_path = os.path.join(src_dir, "main.py")
|
|
151
|
+
with open(main_path, "w") as f:
|
|
152
|
+
f.write(main_content)
|
|
153
|
+
|
|
154
|
+
# 初始化该项目
|
|
155
|
+
original_cwd = os.getcwd()
|
|
156
|
+
try:
|
|
157
|
+
os.chdir(temp_test_dir)
|
|
158
|
+
init_project_if_required(target_dir=temp_test_dir)
|
|
159
|
+
finally:
|
|
160
|
+
os.chdir(original_cwd)
|
|
161
|
+
|
|
162
|
+
# 返回文件内容供测试使用
|
|
163
|
+
return {
|
|
164
|
+
"math_utils": math_utils_content,
|
|
165
|
+
"string_utils": string_utils_content,
|
|
166
|
+
"main": main_content
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
@pytest.fixture
|
|
170
|
+
def sample_conversations(self):
|
|
171
|
+
"""Sample conversations with tool results for testing"""
|
|
172
|
+
# 创建一个长内容用于测试token计数
|
|
173
|
+
long_content = """def hello():
|
|
174
|
+
print('Hello, world!')
|
|
175
|
+
# This is a very long file content that would take up many tokens
|
|
176
|
+
# We want to clean this up to save space in the conversation
|
|
177
|
+
for i in range(100):
|
|
178
|
+
print(f'Line {i}: This is line number {i} with some content')
|
|
179
|
+
return 'done'
|
|
180
|
+
|
|
181
|
+
def calculate_sum(numbers):
|
|
182
|
+
total = 0
|
|
183
|
+
for num in numbers:
|
|
184
|
+
total += num
|
|
185
|
+
return total
|
|
186
|
+
|
|
187
|
+
def process_data(data):
|
|
188
|
+
processed = []
|
|
189
|
+
for item in data:
|
|
190
|
+
if item > 0:
|
|
191
|
+
processed.append(item * 2)
|
|
192
|
+
return processed
|
|
193
|
+
|
|
194
|
+
# More functions to increase token count
|
|
195
|
+
def validate_input(input_data):
|
|
196
|
+
if input_data is None:
|
|
197
|
+
return False
|
|
198
|
+
return True
|
|
199
|
+
|
|
200
|
+
def format_output(result):
|
|
201
|
+
return f"Result: {result}"
|
|
202
|
+
"""
|
|
203
|
+
|
|
204
|
+
return [
|
|
205
|
+
{"role": "system", "content": "You are a helpful assistant."},
|
|
206
|
+
{"role": "user", "content": "Please read a file for me."},
|
|
207
|
+
{"role": "assistant", "content": "I'll read the file for you.\n\n<read_file>\n<path>test.py</path>\n</read_file>"},
|
|
208
|
+
{
|
|
209
|
+
"role": "user",
|
|
210
|
+
"content": f"<tool_result tool_name='ReadFileTool' success='true'><message>File read successfully</message><content>{long_content}</content></tool_result>"
|
|
211
|
+
},
|
|
212
|
+
{"role": "assistant", "content": "I can see the file content. Let me analyze it for you."},
|
|
213
|
+
{"role": "user", "content": "Now please list files in the directory."},
|
|
214
|
+
{"role": "assistant", "content": "I'll list the files for you.\n\n<list_files>\n<path>.</path>\n</list_files>"},
|
|
215
|
+
{
|
|
216
|
+
"role": "user",
|
|
217
|
+
"content": "<tool_result tool_name='ListFilesTool' success='true'><message>Files listed successfully</message><content>['file1.py', 'file2.js', 'file3.md', 'very_long_file_with_many_tokens_that_should_be_cleaned.txt', 'another_file.py', 'config.json', 'readme.md', 'test_data.csv']</content></tool_result>"
|
|
218
|
+
},
|
|
219
|
+
{"role": "assistant", "content": "Here are the files in the directory. Is there anything specific you'd like to do with them?"}
|
|
220
|
+
]
|
|
221
|
+
|
|
222
|
+
def test_initialization(self, mock_args, real_llm):
|
|
223
|
+
"""Test AgenticConversationPruner initialization"""
|
|
224
|
+
pruner = AgenticConversationPruner(args=mock_args, llm=real_llm, conversation_id="test-init-conversation")
|
|
225
|
+
|
|
226
|
+
assert pruner.args == mock_args
|
|
227
|
+
assert pruner.llm == real_llm
|
|
228
|
+
assert hasattr(pruner, 'tool_content_detector')
|
|
229
|
+
assert pruner.replacement_message == "This message has been cleared. If you still want to get this information, you can call the tool again to retrieve it."
|
|
230
|
+
|
|
231
|
+
|
|
232
|
+
|
|
233
|
+
def test_prune_conversations_within_limit(self, pruner, sample_conversations):
|
|
234
|
+
"""Test pruning when conversations are within token limit"""
|
|
235
|
+
# 创建一个短对话来测试不需要修剪的情况
|
|
236
|
+
short_conversations = [
|
|
237
|
+
{"role": "user", "content": "Hello"},
|
|
238
|
+
{"role": "assistant", "content": "Hi there!"}
|
|
239
|
+
]
|
|
240
|
+
|
|
241
|
+
result = pruner.prune_conversations(short_conversations)
|
|
242
|
+
|
|
243
|
+
# Should return original conversations unchanged since they're short
|
|
244
|
+
assert result == short_conversations
|
|
245
|
+
|
|
246
|
+
def test_prune_conversations_exceeds_limit(self, pruner, sample_conversations):
|
|
247
|
+
"""Test pruning when conversations exceed token limit"""
|
|
248
|
+
# 使用真实的token计数进行测试
|
|
249
|
+
# sample_conversations包含了长内容,应该会触发修剪
|
|
250
|
+
|
|
251
|
+
result = pruner.prune_conversations(sample_conversations)
|
|
252
|
+
|
|
253
|
+
# Should have processed the conversations
|
|
254
|
+
assert isinstance(result, list)
|
|
255
|
+
assert len(result) == len(sample_conversations)
|
|
256
|
+
|
|
257
|
+
# Check that tool results were cleaned if token count was high
|
|
258
|
+
# 由于我们使用真实的token计数,可能会也可能不会触发修剪
|
|
259
|
+
# 这取决于实际的token数量
|
|
260
|
+
|
|
261
|
+
# 至少验证结果是有效的对话列表
|
|
262
|
+
for conv in result:
|
|
263
|
+
assert isinstance(conv, dict), "Each conversation should be a dict"
|
|
264
|
+
assert "role" in conv, "Each conversation should have a role"
|
|
265
|
+
assert "content" in conv, "Each conversation should have content"
|
|
266
|
+
|
|
267
|
+
def test_is_tool_result_message(self, pruner):
|
|
268
|
+
"""Test tool result message detection"""
|
|
269
|
+
# Test cases for tool result detection
|
|
270
|
+
test_cases = [
|
|
271
|
+
("<tool_result tool_name='ReadFileTool' success='true'>content</tool_result>", True),
|
|
272
|
+
("<tool_result tool_name=\"ListTool\" success=\"false\">error</tool_result>", True),
|
|
273
|
+
("Regular message without tool result", False),
|
|
274
|
+
("<tool_result>missing tool_name</tool_result>", False),
|
|
275
|
+
("", False),
|
|
276
|
+
("<some_other_tag tool_name='test'>content</some_other_tag>", False)
|
|
277
|
+
]
|
|
278
|
+
|
|
279
|
+
for content, expected in test_cases:
|
|
280
|
+
result = pruner._is_tool_result_message(content)
|
|
281
|
+
assert result == expected, f"Failed for content: {content}"
|
|
282
|
+
|
|
283
|
+
def test_extract_tool_name(self, pruner):
|
|
284
|
+
"""Test tool name extraction from tool results"""
|
|
285
|
+
test_cases = [
|
|
286
|
+
("<tool_result tool_name='ReadFileTool' success='true'>", "ReadFileTool"),
|
|
287
|
+
('<tool_result tool_name="ListFilesTool" success="true">', "ListFilesTool"),
|
|
288
|
+
("<tool_result success='true' tool_name='WriteTool'>", "WriteTool"),
|
|
289
|
+
("<tool_result success='true'>", "unknown"),
|
|
290
|
+
("Not a tool result", "unknown"),
|
|
291
|
+
("", "unknown"),
|
|
292
|
+
("<tool_result tool_name=''>", ""),
|
|
293
|
+
("<tool_result tool_name='Tool With Spaces'>", "Tool With Spaces")
|
|
294
|
+
]
|
|
295
|
+
|
|
296
|
+
for content, expected in test_cases:
|
|
297
|
+
result = pruner._extract_tool_name(content)
|
|
298
|
+
assert result == expected, f"Failed for content: {content}"
|
|
299
|
+
|
|
300
|
+
def test_generate_replacement_message(self, pruner):
|
|
301
|
+
"""Test replacement message generation"""
|
|
302
|
+
test_cases = [
|
|
303
|
+
"ReadFileTool",
|
|
304
|
+
"ListFilesTool",
|
|
305
|
+
"unknown",
|
|
306
|
+
"",
|
|
307
|
+
"CustomTool"
|
|
308
|
+
]
|
|
309
|
+
|
|
310
|
+
for tool_name in test_cases:
|
|
311
|
+
replacement = pruner._generate_replacement_message(tool_name)
|
|
312
|
+
|
|
313
|
+
# Check that replacement contains expected elements
|
|
314
|
+
assert "<tool_result" in replacement
|
|
315
|
+
assert "Content cleared to save tokens" in replacement
|
|
316
|
+
assert pruner.replacement_message in replacement
|
|
317
|
+
|
|
318
|
+
if tool_name and tool_name != "unknown":
|
|
319
|
+
assert f"tool_name='{tool_name}'" in replacement
|
|
320
|
+
|
|
321
|
+
def test_get_cleanup_statistics(self, pruner, sample_conversations):
|
|
322
|
+
"""Test cleanup statistics calculation"""
|
|
323
|
+
# Create pruned conversations (simulate cleaning)
|
|
324
|
+
pruned_conversations = sample_conversations.copy()
|
|
325
|
+
pruned_conversations[3]["content"] = "<tool_result tool_name='ReadFileTool' success='true'><message>Content cleared to save tokens</message><content>This message has been cleared</content></tool_result>"
|
|
326
|
+
|
|
327
|
+
stats = pruner.get_cleanup_statistics(sample_conversations, pruned_conversations)
|
|
328
|
+
|
|
329
|
+
# Verify statistics structure
|
|
330
|
+
assert isinstance(stats, dict), "Stats should be a dictionary"
|
|
331
|
+
assert 'original_tokens' in stats, "Stats should include original_tokens"
|
|
332
|
+
assert 'pruned_tokens' in stats, "Stats should include pruned_tokens"
|
|
333
|
+
assert 'tokens_saved' in stats, "Stats should include tokens_saved"
|
|
334
|
+
assert 'compression_ratio' in stats, "Stats should include compression_ratio"
|
|
335
|
+
assert 'tool_results_cleaned' in stats, "Stats should include tool_results_cleaned"
|
|
336
|
+
assert 'tool_calls_cleaned' in stats, "Stats should include tool_calls_cleaned"
|
|
337
|
+
assert 'total_messages' in stats, "Stats should include total_messages"
|
|
338
|
+
|
|
339
|
+
# Verify that statistics are reasonable
|
|
340
|
+
assert stats['original_tokens'] > 0, "Original tokens should be positive"
|
|
341
|
+
assert stats['pruned_tokens'] >= 0, "Pruned tokens should be non-negative"
|
|
342
|
+
assert stats['tokens_saved'] >= 0, "Tokens saved should be non-negative"
|
|
343
|
+
assert 0 <= stats['compression_ratio'] <= 1, "Compression ratio should be between 0 and 1"
|
|
344
|
+
assert stats['tool_results_cleaned'] >= 0, "Tool results cleaned should be non-negative"
|
|
345
|
+
assert stats['tool_calls_cleaned'] >= 0, "Tool calls cleaned should be non-negative"
|
|
346
|
+
assert stats['total_messages'] == len(sample_conversations), "Total messages should match input"
|
|
347
|
+
|
|
348
|
+
def test_prune_conversations_default_behavior(self, pruner, sample_conversations):
|
|
349
|
+
"""Test pruning with default behavior (no strategy parameter)"""
|
|
350
|
+
# Test simplified interface without strategy parameter
|
|
351
|
+
result = pruner.prune_conversations(sample_conversations)
|
|
352
|
+
assert isinstance(result, list)
|
|
353
|
+
|
|
354
|
+
def test_prune_conversations_empty_list(self, pruner):
|
|
355
|
+
"""Test pruning with empty conversation list"""
|
|
356
|
+
result = pruner.prune_conversations([])
|
|
357
|
+
assert result == []
|
|
358
|
+
|
|
359
|
+
def test_prune_conversations_no_tool_results(self, pruner):
|
|
360
|
+
"""Test pruning conversations without tool results"""
|
|
361
|
+
conversations = [
|
|
362
|
+
{"role": "user", "content": "Hello"},
|
|
363
|
+
{"role": "assistant", "content": "Hi there!"},
|
|
364
|
+
{"role": "user", "content": "How are you?"},
|
|
365
|
+
{"role": "assistant", "content": "I'm doing well, thank you!"}
|
|
366
|
+
]
|
|
367
|
+
|
|
368
|
+
result = pruner.prune_conversations(conversations)
|
|
369
|
+
# Should return original since no tool results to clean
|
|
370
|
+
assert result == conversations
|
|
371
|
+
|
|
372
|
+
def test_progressive_cleanup(self, pruner):
|
|
373
|
+
"""Test that cleanup happens progressively from earliest tool results"""
|
|
374
|
+
# 创建包含多个tool结果的对话,其中包含长内容来触发修剪
|
|
375
|
+
long_result = "# This is a very long result content that should trigger token limit\n" * 100
|
|
376
|
+
|
|
377
|
+
conversations = [
|
|
378
|
+
{"role": "user", "content": "First request"},
|
|
379
|
+
{"role": "user", "content": f"<tool_result tool_name='Tool1'><content>{long_result}</content></tool_result>"},
|
|
380
|
+
{"role": "user", "content": "Second request"},
|
|
381
|
+
{"role": "user", "content": f"<tool_result tool_name='Tool2'><content>{long_result}</content></tool_result>"},
|
|
382
|
+
{"role": "user", "content": "Third request"},
|
|
383
|
+
{"role": "user", "content": f"<tool_result tool_name='Tool3'><content>{long_result}</content></tool_result>"}
|
|
384
|
+
]
|
|
385
|
+
|
|
386
|
+
result = pruner.prune_conversations(conversations)
|
|
387
|
+
|
|
388
|
+
# 验证结果的基本结构
|
|
389
|
+
assert isinstance(result, list), "Result should be a list"
|
|
390
|
+
assert len(result) == len(conversations), "Should preserve conversation count"
|
|
391
|
+
|
|
392
|
+
# 检查是否有tool结果被清理
|
|
393
|
+
for conv in result:
|
|
394
|
+
assert isinstance(conv, dict), "Each conversation should be a dict"
|
|
395
|
+
assert "role" in conv, "Each conversation should have a role"
|
|
396
|
+
assert "content" in conv, "Each conversation should have content"
|
|
397
|
+
|
|
398
|
+
def test_apply_range_pruning_with_pair_preservation(self, mock_args_small_threshold, real_llm):
|
|
399
|
+
"""Test _apply_range_pruning with pair preservation enabled"""
|
|
400
|
+
from unittest.mock import MagicMock, patch
|
|
401
|
+
from autocoder.common.pruner.conversation_message_ids_manager import ConversationMessageIds
|
|
402
|
+
|
|
403
|
+
# 创建包含user/assistant对话的测试数据
|
|
404
|
+
conversations = [
|
|
405
|
+
{"role": "system", "content": "You are a helpful assistant.", "message_id": "12345678abc"},
|
|
406
|
+
{"role": "user", "content": "Hello!", "message_id": "23456789def"}, # user
|
|
407
|
+
{"role": "assistant", "content": "Hi there!", "message_id": "34567890ghi"}, # assistant
|
|
408
|
+
{"role": "user", "content": "Can you help?", "message_id": "45678901jkl"}, # user
|
|
409
|
+
{"role": "assistant", "content": "Of course!", "message_id": "56789012mno"}, # assistant
|
|
410
|
+
{"role": "user", "content": "Thank you!", "message_id": "67890123pqr"} # user (single)
|
|
411
|
+
]
|
|
412
|
+
|
|
413
|
+
# 尝试只删除一个user消息,但由于pair preservation,应该连同对应的assistant也被删除
|
|
414
|
+
message_ids_to_delete = ["23456789"] # 只删除第一个user消息
|
|
415
|
+
conversation_id = "test_conversation_pair_123"
|
|
416
|
+
|
|
417
|
+
# 创建带有pair preservation的配置
|
|
418
|
+
mock_conversation_message_ids = ConversationMessageIds(
|
|
419
|
+
conversation_id=conversation_id,
|
|
420
|
+
message_ids=message_ids_to_delete,
|
|
421
|
+
created_at="2024-01-01T00:00:00",
|
|
422
|
+
updated_at="2024-01-01T00:00:00",
|
|
423
|
+
description="Test pair preservation",
|
|
424
|
+
preserve_pairs=True # 启用成对保护
|
|
425
|
+
)
|
|
426
|
+
|
|
427
|
+
pruner = AgenticConversationPruner(args=mock_args_small_threshold, llm=real_llm, conversation_id=conversation_id)
|
|
428
|
+
|
|
429
|
+
with patch.object(pruner.message_ids_api, 'get_conversation_message_ids') as mock_get_message_ids:
|
|
430
|
+
mock_get_message_ids.return_value = mock_conversation_message_ids
|
|
431
|
+
|
|
432
|
+
result = pruner.prune_conversations(conversations)
|
|
433
|
+
|
|
434
|
+
# 验证结果
|
|
435
|
+
assert isinstance(result, list), "Result should be a list"
|
|
436
|
+
|
|
437
|
+
# 由于pair preservation,第一个user和其对应的assistant都应该被删除
|
|
438
|
+
result_message_ids = [conv.get("message_id", "")[:8] for conv in result]
|
|
439
|
+
|
|
440
|
+
# 验证成对删除是否正确执行
|
|
441
|
+
# 注意:具体的行为取决于ConversationMessageIdsPruner的实现
|
|
442
|
+
# 这里主要验证range pruning确实被触发了
|
|
443
|
+
stats = pruner.get_pruning_statistics()
|
|
444
|
+
assert stats["range_pruning"]["applied"] == True, "Range pruning with pair preservation should be applied"
|
|
445
|
+
|
|
446
|
+
print(f"✅ Range pruning with pair preservation test passed!")
|
|
447
|
+
print(f" Original messages: {len(conversations)}")
|
|
448
|
+
print(f" Messages after range pruning: {len(result)}")
|
|
449
|
+
print(f" Preserve pairs enabled: {mock_conversation_message_ids.preserve_pairs}")
|
|
450
|
+
|
|
451
|
+
def test_apply_range_pruning_no_conversation_id(self, mock_args_small_threshold, real_llm):
|
|
452
|
+
"""Test that AgenticConversationPruner requires conversation_id parameter"""
|
|
453
|
+
# 测试当没有提供 conversation_id 时构造函数应该抛出 ValueError
|
|
454
|
+
with pytest.raises(ValueError, match="conversation_id is required"):
|
|
455
|
+
pruner = AgenticConversationPruner(args=mock_args_small_threshold, llm=real_llm)
|
|
456
|
+
|
|
457
|
+
print("✅ Test passed: AgenticConversationPruner correctly requires conversation_id parameter")
|
|
458
|
+
|
|
459
|
+
def test_apply_range_pruning_no_message_ids_config(self, mock_args_small_threshold, real_llm):
|
|
460
|
+
"""Test that _apply_range_pruning is skipped when no message IDs configuration exists"""
|
|
461
|
+
from unittest.mock import patch
|
|
462
|
+
|
|
463
|
+
# 添加长内容以确保超过token阈值,从而触发裁剪流程
|
|
464
|
+
long_content = "This is a very long message content that should definitely exceed the token threshold. " * 10
|
|
465
|
+
|
|
466
|
+
conversations = [
|
|
467
|
+
{"role": "user", "content": f"Hello. {long_content}"},
|
|
468
|
+
{"role": "assistant", "content": f"Hi there! {long_content}"}
|
|
469
|
+
]
|
|
470
|
+
|
|
471
|
+
conversation_id = "test_conversation_no_config"
|
|
472
|
+
pruner = AgenticConversationPruner(args=mock_args_small_threshold, llm=real_llm, conversation_id=conversation_id)
|
|
473
|
+
|
|
474
|
+
# Mock返回None,表示没有找到消息ID配置
|
|
475
|
+
with patch.object(pruner.message_ids_api, 'get_conversation_message_ids') as mock_get_message_ids:
|
|
476
|
+
mock_get_message_ids.return_value = None
|
|
477
|
+
|
|
478
|
+
result = pruner.prune_conversations(conversations)
|
|
479
|
+
|
|
480
|
+
# 验证mock被调用
|
|
481
|
+
mock_get_message_ids.assert_called_once_with(conversation_id)
|
|
482
|
+
|
|
483
|
+
# 由于我们使用了长内容且阈值很小,可能会添加cleanup提示消息
|
|
484
|
+
# 验证原始消息仍然存在(可能在最后添加了一个提示消息)
|
|
485
|
+
assert len(result) >= len(conversations), "Result should have at least the original conversations"
|
|
486
|
+
|
|
487
|
+
# 验证原始消息的内容没有改变
|
|
488
|
+
for i, original_conv in enumerate(conversations):
|
|
489
|
+
assert result[i]["role"] == original_conv["role"], f"Role should match for message {i}"
|
|
490
|
+
assert result[i]["content"] == original_conv["content"], f"Content should match for message {i}"
|
|
491
|
+
|
|
492
|
+
# 验证统计信息
|
|
493
|
+
stats = pruner.get_pruning_statistics()
|
|
494
|
+
assert stats["range_pruning"]["applied"] == False, "Range pruning should not be applied when no config"
|
|
495
|
+
|
|
496
|
+
print(f"✅ No message IDs config test passed!")
|
|
497
|
+
|
|
498
|
+
def test_tool_call_content_detection(self, pruner):
|
|
499
|
+
"""Test tool call content detection and cleanup"""
|
|
500
|
+
# Test conversations with tool calls
|
|
501
|
+
conversations = [
|
|
502
|
+
{"role": "user", "content": "Please write a file"},
|
|
503
|
+
{
|
|
504
|
+
"role": "assistant",
|
|
505
|
+
"content": """I'll write the file for you.
|
|
506
|
+
|
|
507
|
+
<write_to_file>
|
|
508
|
+
<path>test.py</path>
|
|
509
|
+
<content>print("Hello, World!")
|
|
510
|
+
print("This is a test file")
|
|
511
|
+
print("It contains multiple lines")
|
|
512
|
+
print("And some more content to exceed the length limit")
|
|
513
|
+
print("Even more content to make it longer")
|
|
514
|
+
print("And yet more content to ensure it's over 500 characters")
|
|
515
|
+
</content>
|
|
516
|
+
</write_to_file>
|
|
517
|
+
|
|
518
|
+
File written successfully."""
|
|
519
|
+
}
|
|
520
|
+
]
|
|
521
|
+
|
|
522
|
+
# Check that tool call content is detected
|
|
523
|
+
assert pruner.tool_content_detector.is_tool_call_content(conversations[1]["content"])
|
|
524
|
+
|
|
525
|
+
# Test replacement
|
|
526
|
+
original_content = conversations[1]["content"]
|
|
527
|
+
new_content, replaced = pruner.tool_content_detector.replace_tool_content(
|
|
528
|
+
original_content, max_content_length=100
|
|
529
|
+
)
|
|
530
|
+
|
|
531
|
+
assert replaced, "Tool call content should be replaced when it exceeds length limit"
|
|
532
|
+
assert len(new_content) < len(original_content), "New content should be shorter"
|
|
533
|
+
assert "Content cleared to save tokens" in new_content, "Should contain replacement message"
|
|
534
|
+
|
|
535
|
+
def test_combined_cleanup_flow(self, pruner):
|
|
536
|
+
"""Test the combined cleanup flow (tool outputs + tool calls)"""
|
|
537
|
+
# 创建包含长内容的对话来触发修剪
|
|
538
|
+
long_file_content = "print('This is a very long file content that should be cleaned up to save tokens in the conversation history')\n" * 50
|
|
539
|
+
|
|
540
|
+
conversations = [
|
|
541
|
+
{"role": "user", "content": "Please write a file"},
|
|
542
|
+
{
|
|
543
|
+
"role": "assistant",
|
|
544
|
+
"content": f"""<write_to_file>
|
|
545
|
+
<path>test.py</path>
|
|
546
|
+
<content>{long_file_content}</content>
|
|
547
|
+
</write_to_file>"""
|
|
548
|
+
},
|
|
549
|
+
{
|
|
550
|
+
"role": "user",
|
|
551
|
+
"content": f"""<tool_result tool_name='write_to_file' success='true'>
|
|
552
|
+
<message>File written successfully</message>
|
|
553
|
+
<content>File has been written with the following content:
|
|
554
|
+
{long_file_content}
|
|
555
|
+
</content>
|
|
556
|
+
</tool_result>"""
|
|
557
|
+
}
|
|
558
|
+
]
|
|
559
|
+
|
|
560
|
+
result = pruner.prune_conversations(conversations)
|
|
561
|
+
|
|
562
|
+
# 验证结果的基本结构
|
|
563
|
+
assert isinstance(result, list), "Result should be a list"
|
|
564
|
+
assert len(result) == len(conversations), "Should preserve conversation count"
|
|
565
|
+
|
|
566
|
+
# 验证每个对话都有基本的结构
|
|
567
|
+
for conv in result:
|
|
568
|
+
assert isinstance(conv, dict), "Each conversation should be a dict"
|
|
569
|
+
assert "role" in conv, "Each conversation should have a role"
|
|
570
|
+
assert "content" in conv, "Each conversation should have content"
|
|
571
|
+
|
|
572
|
+
def test_edge_cases(self, pruner):
|
|
573
|
+
"""Test various edge cases"""
|
|
574
|
+
# Test with None content
|
|
575
|
+
assert not pruner._is_tool_result_message(None)
|
|
576
|
+
|
|
577
|
+
# Test with malformed tool result
|
|
578
|
+
malformed = "<tool_result tool_name='Test' incomplete"
|
|
579
|
+
assert pruner._extract_tool_name(malformed) == "Test"
|
|
580
|
+
|
|
581
|
+
# Test with special characters in tool name
|
|
582
|
+
special_chars = "<tool_result tool_name='Tool-With_Special.Chars123' success='true'>"
|
|
583
|
+
assert pruner._extract_tool_name(special_chars) == "Tool-With_Special.Chars123"
|
|
584
|
+
|
|
585
|
+
|
|
586
|
+
class TestAgenticConversationPrunerIntegration:
|
|
587
|
+
"""Integration tests for AgenticConversationPruner"""
|
|
588
|
+
|
|
589
|
+
@pytest.fixture
|
|
590
|
+
def real_args(self):
|
|
591
|
+
"""Create real AutoCoderArgs for integration testing"""
|
|
592
|
+
return AutoCoderArgs(
|
|
593
|
+
source_dir=".",
|
|
594
|
+
conversation_prune_safe_zone_tokens=50000,
|
|
595
|
+
context_prune=True,
|
|
596
|
+
context_prune_strategy="extract",
|
|
597
|
+
context_prune_sliding_window_size=10,
|
|
598
|
+
context_prune_sliding_window_overlap=2,
|
|
599
|
+
query="请帮我分析这些代码文件"
|
|
600
|
+
)
|
|
601
|
+
|
|
602
|
+
@pytest.fixture
|
|
603
|
+
def real_llm(self):
|
|
604
|
+
"""创建真实的LLM对象"""
|
|
605
|
+
llm = get_llm("v3_chat", product_mode="lite")
|
|
606
|
+
return llm
|
|
607
|
+
|
|
608
|
+
@pytest.fixture
|
|
609
|
+
def temp_test_dir(self, tmp_path):
|
|
610
|
+
"""Create a temporary test directory"""
|
|
611
|
+
return str(tmp_path)
|
|
612
|
+
|
|
613
|
+
@pytest.fixture
|
|
614
|
+
def sample_file_sources(self):
|
|
615
|
+
"""Sample file sources for testing"""
|
|
616
|
+
return {
|
|
617
|
+
'math_utils': '''def add(a, b):
|
|
618
|
+
"""加法函数"""
|
|
619
|
+
return a + b
|
|
620
|
+
|
|
621
|
+
def subtract(a, b):
|
|
622
|
+
"""减法函数"""
|
|
623
|
+
return a - b
|
|
624
|
+
|
|
625
|
+
def multiply(a, b):
|
|
626
|
+
"""乘法函数"""
|
|
627
|
+
return a * b''',
|
|
628
|
+
'main': '''from utils.math_utils import add, subtract, multiply
|
|
629
|
+
|
|
630
|
+
def main():
|
|
631
|
+
"""主函数"""
|
|
632
|
+
result1 = add(10, 5)
|
|
633
|
+
result2 = subtract(10, 5)
|
|
634
|
+
result3 = multiply(10, 5)
|
|
635
|
+
print(f"结果: {result1}, {result2}, {result3}")
|
|
636
|
+
|
|
637
|
+
if __name__ == "__main__":
|
|
638
|
+
main()'''
|
|
639
|
+
}
|
|
640
|
+
|
|
641
|
+
def test_integration_real_conversation_scenarios(self, real_args, real_llm):
|
|
642
|
+
"""Test with realistic conversation scenario"""
|
|
643
|
+
# 简化测试,去掉复杂的项目依赖
|
|
644
|
+
conversation_id = "integration-test-conversation"
|
|
645
|
+
pruner = AgenticConversationPruner(args=real_args, llm=real_llm, conversation_id=conversation_id)
|
|
646
|
+
|
|
647
|
+
# Create a realistic conversation with large tool outputs
|
|
648
|
+
large_file_content = "# " + "Very long file content " * 500 # 减少内容大小
|
|
649
|
+
|
|
650
|
+
conversations = [
|
|
651
|
+
{"role": "system", "content": "You are a helpful coding assistant."},
|
|
652
|
+
{"role": "user", "content": "Can you read the main.py file and analyze it?"},
|
|
653
|
+
{"role": "assistant", "content": "I'll read the file for you.\n\n<read_file>\n<path>main.py</path>\n</read_file>"},
|
|
654
|
+
{
|
|
655
|
+
"role": "user",
|
|
656
|
+
"content": f"<tool_result tool_name='ReadFileTool' success='true'><message>File read successfully</message><content>{large_file_content}</content></tool_result>"
|
|
657
|
+
},
|
|
658
|
+
{"role": "assistant", "content": "I can see this is a large Python file. Let me analyze its structure..."},
|
|
659
|
+
{"role": "user", "content": "Now can you list all Python files in the directory?"},
|
|
660
|
+
{"role": "assistant", "content": "I'll list the Python files.\n\n<list_files>\n<path>.</path>\n<pattern>*.py</pattern>\n</list_files>"},
|
|
661
|
+
{
|
|
662
|
+
"role": "user",
|
|
663
|
+
"content": f"<tool_result tool_name='ListFilesTool' success='true'><message>Files listed</message><content>{json.dumps(['file' + str(i) + '.py' for i in range(50)])}</content></tool_result>"
|
|
664
|
+
}
|
|
665
|
+
]
|
|
666
|
+
|
|
667
|
+
result = pruner.prune_conversations(conversations)
|
|
668
|
+
|
|
669
|
+
# Verify the result is valid
|
|
670
|
+
assert isinstance(result, list)
|
|
671
|
+
assert len(result) == len(conversations)
|
|
672
|
+
|
|
673
|
+
# Verify that some cleanup occurred
|
|
674
|
+
stats = pruner.get_cleanup_statistics(conversations, result)
|
|
675
|
+
assert isinstance(stats, dict)
|
|
676
|
+
assert all(key in stats for key in ['original_tokens', 'pruned_tokens', 'tokens_saved', 'compression_ratio', 'tool_results_cleaned', 'total_messages'])
|
|
677
|
+
|
|
678
|
+
def test_integration_mixed_content_pruning(self, real_args, real_llm, sample_file_sources):
|
|
679
|
+
"""Test AgenticConversationPruner with real file sources"""
|
|
680
|
+
# 简化测试,去掉复杂的项目依赖
|
|
681
|
+
conversation_id = "mixed-content-test-conversation"
|
|
682
|
+
pruner = AgenticConversationPruner(args=real_args, llm=real_llm, conversation_id=conversation_id)
|
|
683
|
+
|
|
684
|
+
# Create conversations with file content from sample_file_sources
|
|
685
|
+
conversations = [
|
|
686
|
+
{"role": "system", "content": "You are a helpful coding assistant."},
|
|
687
|
+
{"role": "user", "content": "请分析这些数学工具函数"},
|
|
688
|
+
{"role": "assistant", "content": "我会分析这些数学工具函数。\n\n<read_file>\n<path>src/utils/math_utils.py</path>\n</read_file>"},
|
|
689
|
+
{
|
|
690
|
+
"role": "user",
|
|
691
|
+
"content": f"<tool_result tool_name='ReadFileTool' success='true'><message>File read successfully</message><content>{sample_file_sources['math_utils']}</content></tool_result>"
|
|
692
|
+
},
|
|
693
|
+
{"role": "assistant", "content": "我看到这个文件包含了基本的数学运算函数。让我也看看主程序文件。\n\n<read_file>\n<path>src/main.py</path>\n</read_file>"},
|
|
694
|
+
{
|
|
695
|
+
"role": "user",
|
|
696
|
+
"content": f"<tool_result tool_name='ReadFileTool' success='true'><message>File read successfully</message><content>{sample_file_sources['main']}</content></tool_result>"
|
|
697
|
+
},
|
|
698
|
+
{"role": "assistant", "content": "很好,主程序使用了数学工具函数。这些函数实现了基本的数学运算。"}
|
|
699
|
+
]
|
|
700
|
+
|
|
701
|
+
# 测试pruning功能
|
|
702
|
+
result = pruner.prune_conversations(conversations)
|
|
703
|
+
|
|
704
|
+
# 验证结果
|
|
705
|
+
assert isinstance(result, list), "Result should be a list"
|
|
706
|
+
assert len(result) == len(conversations), "Should preserve conversation count"
|
|
707
|
+
|
|
708
|
+
# 验证每个对话的基本结构
|
|
709
|
+
for i, conv in enumerate(result):
|
|
710
|
+
assert isinstance(conv, dict), f"Conversation {i} should be a dict"
|
|
711
|
+
assert "role" in conv, f"Conversation {i} should have a role"
|
|
712
|
+
assert "content" in conv, f"Conversation {i} should have content"
|
|
713
|
+
assert isinstance(conv["content"], str), f"Conversation {i} content should be a string"
|
|
714
|
+
|
|
715
|
+
# 验证统计信息
|
|
716
|
+
stats = pruner.get_cleanup_statistics(conversations, result)
|
|
717
|
+
assert isinstance(stats, dict), "Stats should be a dictionary"
|
|
718
|
+
assert stats["total_messages"] == len(conversations), "Should count all messages"
|
|
719
|
+
assert stats["original_tokens"] > 0, "Should have original tokens"
|
|
720
|
+
assert stats["pruned_tokens"] >= 0, "Should have pruned tokens"
|
|
721
|
+
|
|
722
|
+
print(f"测试完成 - 原始tokens: {stats['original_tokens']}, 处理后tokens: {stats['pruned_tokens']}")
|
|
723
|
+
|
|
724
|
+
|
|
725
|
+
# Parametrized tests for comprehensive coverage
|
|
726
|
+
class TestParametrized:
|
|
727
|
+
"""Parametrized tests for comprehensive coverage"""
|
|
728
|
+
|
|
729
|
+
@pytest.mark.parametrize("tool_name,expected", [
|
|
730
|
+
("ReadFileTool", "ReadFileTool"),
|
|
731
|
+
("ListFilesTool", "ListFilesTool"),
|
|
732
|
+
("WriteTool", "WriteTool"),
|
|
733
|
+
("", ""),
|
|
734
|
+
("Tool_With_Underscores", "Tool_With_Underscores"),
|
|
735
|
+
("Tool-With-Hyphens", "Tool-With-Hyphens"),
|
|
736
|
+
("Tool123", "Tool123"),
|
|
737
|
+
])
|
|
738
|
+
def test_tool_name_extraction_parametrized(self, tool_name, expected):
|
|
739
|
+
"""Parametrized test for tool name extraction"""
|
|
740
|
+
# 创建真实的参数和LLM对象
|
|
741
|
+
args = AutoCoderArgs(
|
|
742
|
+
source_dir=".",
|
|
743
|
+
conversation_prune_safe_zone_tokens=1000,
|
|
744
|
+
context_prune=True,
|
|
745
|
+
context_prune_strategy="extract",
|
|
746
|
+
query="测试查询"
|
|
747
|
+
)
|
|
748
|
+
llm = get_llm("v3_chat", product_mode="lite")
|
|
749
|
+
assert llm is not None, "LLM should not be None"
|
|
750
|
+
conversation_id = "parametrized-test-conversation"
|
|
751
|
+
pruner = AgenticConversationPruner(args, llm, conversation_id=conversation_id)
|
|
752
|
+
content = f"<tool_result tool_name='{tool_name}' success='true'>"
|
|
753
|
+
result = pruner._extract_tool_name(content)
|
|
754
|
+
assert result == expected
|
|
755
|
+
|
|
756
|
+
@pytest.mark.parametrize("content,expected", [
|
|
757
|
+
("<tool_result tool_name='Test' success='true'>content</tool_result>", True),
|
|
758
|
+
("<tool_result tool_name=\"Test\" success=\"true\">content</tool_result>", True),
|
|
759
|
+
("Regular message", False),
|
|
760
|
+
("<tool_result>no tool_name</tool_result>", False),
|
|
761
|
+
("", False),
|
|
762
|
+
("<other_tag tool_name='test'>content</other_tag>", False),
|
|
763
|
+
])
|
|
764
|
+
def test_tool_result_detection_parametrized(self, content, expected):
|
|
765
|
+
"""Parametrized test for tool result detection"""
|
|
766
|
+
# 创建真实的参数和LLM对象
|
|
767
|
+
args = AutoCoderArgs(
|
|
768
|
+
source_dir=".",
|
|
769
|
+
conversation_prune_safe_zone_tokens=1000,
|
|
770
|
+
context_prune=True,
|
|
771
|
+
context_prune_strategy="extract",
|
|
772
|
+
query="测试查询"
|
|
773
|
+
)
|
|
774
|
+
llm = get_llm("v3_chat", product_mode="lite")
|
|
775
|
+
assert llm is not None, "LLM should not be None"
|
|
776
|
+
conversation_id = "tool-result-detection-test"
|
|
777
|
+
pruner = AgenticConversationPruner(args, llm, conversation_id=conversation_id)
|
|
778
|
+
result = pruner._is_tool_result_message(content)
|
|
779
|
+
assert result == expected
|
|
780
|
+
|
|
781
|
+
|
|
782
|
+
if __name__ == "__main__":
|
|
783
|
+
# Run tests with pytest
|
|
784
|
+
pytest.main([__file__, "-v"])
|